//引入vue-router路由插件
import VueRouter from "vue-router";
//引入Vue
import Vue from "vue";
import routes from "./routes";
//使用插件
Vue.use(VueRouter);
//引入store
import store from "@/store";

import Home from '@/pages/Home';
import Register from '@/pages/Register';
import Search from '@/pages/Search';
import Login from '@/pages/Login';

//需要重写VueRouter.prototype原型对象身上的push|replace方法
//先把VueRouter.prototype身上的push|replace方法进行保存一份

let originPush = VueRouter.prototype.push;
let originReplace = VueRouter.prototype.replace;

// 简易版: 重写 push 或 replace方法
/*
VueRouter.prototype.push = function () {
    alert('重写 push 或 replace方法')
};
*/


/*
* call、apply的区别
*
* 相同点: 都可以调用函数一次, 都可以篡改函数的上下文
*
* 不同点: call传递参数用逗号隔开; 而apply传递参数用数组。
*
* */

VueRouter.prototype.push = function (location, resolve, reject) {
    if (resolve && reject){
        // 如果不加上call, 则是window对象调用了函数。
        originPush.call(this, location, resolve, reject);
    } else {
        // 手动加上回调函数
        originPush.call(this, location, ()=>{}, ()=>{});
    }
};


VueRouter.prototype.replace = function (location, resolve, reject) {
    if (resolve && reject){
        // 如果不加上call, 则是window对象调用了函数。
        originReplace.call(this, location, resolve, reject);
    } else {
        // 手动加上回调函数
        originReplace.call(this, location, ()=>{}, ()=>{});
    }
};


//对外暴露VueRouter类的实例
let router = new VueRouter({
    //配置路由
    //第一:路径的前面需要有/(不是二级路由)
    //路径中单词都是小写的
    //component右侧V别给我加单引号【字符串：组件是对象（VueComponent类的实例）】
    routes,
    //滚动行为
    scrollBehavior(to, from, savedPosition) {
        //返回的这个y=0，代表的滚动条在最上方
        return { y: 0 };
    },
});




//全局守卫：前置守卫（在路由跳转之间进行判断）
router.beforeEach(async (to, from, next) => {
    //to:获取到要跳转到的路由信息
    //from：获取到从哪个路由跳转过来来的信息
    //next: next() 放行  next(path) 放行
    //方便测试 统一放行
    //  next();
    //获取仓库中的token-----可以确定用户是登录了
    let token  = store.state.user.token;
    let name = store.state.user.userInfo.name;
    //用户登录了
    if(token){
        //已经登录而且还想去登录------不行
        if(to.path==="/login"||to.path==='/register'){
            next('/');
        }else{
            //已经登陆了,访问的是非登录与注册
            //登录了且拥有用户信息放行
            if(name){
                next();
            }else{
                //登陆了且没有用户信息
                //在路由跳转之前获取用户信息且放行
                try {
                    await store.dispatch('getUserInfo');
                    next();
                } catch (error) {
                    //token失效从新登录
                    await store.dispatch('userLogout');
                    next('/login')
                }
            }
        }
    }else{
        //未登录：不能去交易相关、不能去支付相关【pay|paysuccess】、不能去个人中心
        //未登录去上面这些路由-----登录
        let toPath = to.path;
        // 不允许放过的关键词
        let notAllowInKeyword = ['/trade', '/pay', '/center'];
        // some是: 若有一个元素满足指定条件（由于后面的函数提供），则表达式返回true , 剩余的元素不会再执行检测。
        console.log('是否允许通过', notAllowInKeyword.some(itemStr => toPath.includes(itemStr)))
        if( notAllowInKeyword.some(itemStr => toPath.includes(itemStr)) ){
            //把未登录的时候向去而没有去成的信息，存储于地址栏中【路由】
            next('/login?redirect='+toPath);
        }else{
            //去的不是上面这些路由（home|search|shopCart）---放行
            next();
        }

    }
});

export default router;
